<template>
  <el-dialog :title="title" :visible="showDialog" @close="btnCancel">
    <!-- 表单组件  el-form   label-width设置label的宽度   -->
    <!-- 匿名插槽 -->
    <el-form ref="deptForm" label-width="120px" :model="formData" :rules="rules">
      <el-form-item label="部门名称" prop="name">
        <el-input
          v-model="formData.name"
          style="width: 80%"
          placeholder="1-50个字符"
        />
      </el-form-item>
      <el-form-item label="部门编码" prop="code">
        <el-input
          v-model="formData.code"
          style="width: 80%"
          placeholder="1-50个字符"
        />
      </el-form-item>
      <el-form-item label="部门负责人" prop="manager">
        <el-select
          v-model="formData.manager"
          style="width: 80%"
          placeholder="请选择"
          @focus="getEmployeeSimple"
        >
          <el-option
            v-for="item in peoples"
            :key="item.id"
            :label="item.username"
            :value="item.username"
          />
        </el-select>
      </el-form-item>
      <el-form-item label="部门介绍" prop="introduce">
        <el-input
          v-model="formData.introduce"
          style="width: 80%"
          placeholder="1-300个字符"
          type="textarea"
          :rows="3"
        />
      </el-form-item>
    </el-form>
    <!-- el-dialog有专门放置底部操作栏的 插槽  具名插槽 -->
    <el-row slot="footer" type="flex" justify="center">
      <!-- 列被分为24 -->
      <el-col :span="6">
        <el-button type="primary" size="small" @click="btnOk">确定</el-button>
        <el-button size="small" @click="btnCancel">取消</el-button>
      </el-col>
    </el-row>
  </el-dialog>
</template>

<script>
import { addDepartments, getDepartments, getDepartDetail, updateDepartments } from '@/api/departments'
import { getEmployeeSimple } from '@/api/employees'
export default {
  props: {
    showDialog: {
      type: Boolean,
      default: false
    },
    treeNode: {
      type: Object,
      required: true
    }
  },
  data() {
    const checkNameRepeat = async(rule, value, callback) => {
      // 成功 直接调用 callback()
      // 失败 调用 callback(new Error('错误原因'))
      // 这里要校验当前输入名, 跟同一个父部门下的其他子部门没有重复
      // 1. 获取所有部门数据
      const res = await getDepartments()
      // 2. 遍历所有的部门, 如果有一个部门 pid 跟我当前需要添加的父部门相同 item.pid === treeNode.id
      // 3. 而且 item.name === value 就证明出错了

      // 本来拿回来的是所有部门, 可以先用筛选, 筛选出属于当前父部门的数据
      // 数组的过滤器可以实现, 过滤器的原理, 就是遍历所有的数据, 每个数据都执行一下回调
      // 如果回调返回 true 就保留, 否则抛弃
      // res.filter(item => item.pid === this.treeNode.id)
      // 然后在这个基础上, 判断是否有同名
      // 使用 some 方法, 可以判断一个数组当中, 是否存在某些元素符合规定
      // 如果符合则返回 true 否则返回 false

      // 现在可能有编辑状态
      // 根据编辑或者新增, 应该是不同的校验规则才对
      let isRepeat
      if (this.formData.id) {
        // 编辑
        // 编辑时, 当前的 treeNode 其实就是被编辑的数据
        // 1. 找到跟当前 treeNode.pid 相同的部门才是我的同级部门
        // 2. 找的时候将自己排除在外 item.id !== treeNode.id
        isRepeat = res.depts
          .filter(item => item.pid === this.treeNode.pid && item.id !== this.treeNode.id)
        // 3.查找有没有同名
          .some(item => item.name === value)
      } else {
        // 新增
        isRepeat = res.depts
          .filter((item) => item.pid === this.treeNode.id)
          .some((item) => item.name === value)
      }

      if (isRepeat) {
        callback(new Error('同一部门下, 不能有两个同名的子部门'))
      } else {
        callback()
      }
    }
    const checkCodeRepeat = async(rule, value, callback) => {
      // 1. 获取所有部门数据
      const res = await getDepartments()

      let isRepeat
      // 2. 遍历所有部门, 如果有一个或以上的 code 跟当前用户输入的 value 相同, 则需要报错
      // 如果要兼容编辑状况, 那么一旦发现编辑的情况, 校验时应该将自己排除在外
      if (this.formData.id) {
        // 编辑状况
        isRepeat = res.depts.some((item) => item.id !== this.treeNode.id && item.code === value && value)
      } else {
        // 新建状况
        isRepeat = res.depts.some((item) => item.code === value && value)
      }

      if (isRepeat) {
        callback(new Error('部门编码必须是唯一值, 不能重复'))
      } else {
        callback()
      }
    }
    return {
      formData: {
        name: '',
        code: '',
        manager: '',
        introduce: ''
      },
      rules: {
        name: [
          { required: true, trigger: 'blur', message: '部门名称不能为空' },
          {
            min: 1,
            max: 50,
            trigger: 'blur',
            message: '部门名称要求1到50个字符'
          },
          { trigger: 'blur', validator: checkNameRepeat }
        ],
        code: [
          { required: true, trigger: 'blur', message: '部门代码不能为空' },
          {
            min: 1,
            max: 50,
            trigger: 'blur',
            message: '部门代码要求1到50个字符'
          },
          { trigger: 'blur', validator: checkCodeRepeat }
        ],
        manager: [
          { required: true, trigger: 'blur', message: '负责人不能为空' }
        ],
        introduce: [
          { required: true, trigger: 'blur', message: '介绍不能为空' },
          {
            min: 1,
            max: 300,
            trigger: 'blur',
            message: '介绍要求1到300个字符'
          }
        ]
      },
      peoples: []
    }
  },
  computed: {
    title() {
      return this.formData.id ? '编辑部门' : '新增部门'
    }
  },
  created() {
    console.log('创建弹窗组件')
  },
  methods: {
    async getEmployeeSimple() {
      this.peoples = await getEmployeeSimple()
    },
    async btnOk() {
      try {
        await this.$refs.deptForm.validate()

        if (this.formData.id) {
          // 编辑场景
          await updateDepartments(this.formData)
        } else {
          // 新增场景
          // 1. 校验
          // 2. 发请求
          await addDepartments({ ...this.formData, pid: this.treeNode.id })
        }

        // 3. 更新数据
        this.$emit('addDepts')
        // 4. 关闭弹窗
        // 如果在子组件内想要修改父组件传过来的 props 值
        // 有一个固定写法
        // this.$emit(update:props名字, 想要修改的值)
        this.$emit('update:showDialog', false)

        this.$message.success('操作成功')
      } catch (error) {
        console.log(error)
      }
    },
    btnCancel() {
      // 兼容编辑状态, 对于回显的, 非表单绑定数据进行一个清理
      this.formData = {
        name: '',
        code: '',
        manager: '',
        introduce: ''
      }

      // 重置校验字段
      // form 组件有一个方法 resetFields 可以讲所有的当前 formData 绑定的数据
      // 和现在正在显示的错误信息全部清理掉
      this.$refs.deptForm.resetFields()
      // 关闭弹窗
      this.$emit('update:showDialog', false)
    },
    async getDepartDetail(id) {
      console.log('需要在编辑弹起被调用')
      this.formData = await getDepartDetail(id)
    }
  }
}
</script>

<style>
</style>
